

## Faculté Polytechnique



## Rapport de projet de Hardware Software

Ma1 Ingenieur Civil Electricien

Année académique 2020-2021

May 2021

Julio Cesar Betheu Talla et Jaurès Anou Koudjou



Sous la supervision de M.Carlos Valderama



# Table des matières

| Introdu | uction                | 1  |
|---------|-----------------------|----|
| 0.1     | PWM Generator 50M.vhd | 1  |
| 0.2     | Sofware               | 6  |
| 0.3     | TEST BENCH            | 9  |
| 0.4     | Simulation            | 12 |
|         | 0.4.1 DUTY_INCREASE   | 13 |
|         | 0.4.2 DUTY_DECREASE   | 14 |
| 0.5     | Conclusion            | 16 |

### Introduction

The goal of this project is to control a DC motor by pwm. To achieve this, a code to generate a pwm signal will be written in VHDL language and then implemented on the DE1 ALTERA board.

#### - PWM

A PWM signal is a periodic signal. In binary, it varies between '1' and '0'. The value of a pwm is defined by an important parameter, the duty cycle. This is the ratio of the time the signal is at '1' to the time of a period. To control the DC motor, the average value of the pwm signal is used, which depends directly on the duty cycle. The higher the value, the higher the average value.

- FPGA The FPGA card is produced by INTEL. It is composed of one RS-232 port, one PS/2 port, one USB port, one VGA output, 72 cyclone II I/O pins, 4 push buttons and 10 switches. 10 red LEDs and a seven-segment display are present on the board. It has 512 Kbytes of SRAM memory and 8 Mbytes of dynamic SDRAM memory. It has 3 internal clocks  $(24 \mathrm{MHz} - 27 \mathrm{MHz} - 50 \mathrm{MHz})$  and allows the use of an external clock thanks to the SMA input.

#### 0.1 PWM Generator 50M.vhd

- Definition of the entity's entry and exit ports

The first thing to do when writing a vhdl file is to define its entity with its input and output ports. In our case, the entity is called PWM Generator 50M and has 4 input and output ports. The first input port corresponds to the 50MHz clock, this clock will be the reference clock in the system and will be brought as input by our DE1 Altera card. Then, the 3 other input ports correspond to the activation of 3 different buttons on the board. The first button DUTY\_INCREASE will increase the Duty Cycle by 10%, the second DUTY\_DECREASE will decrease the Duty Cycle by 10% and finally the third button direction H will allow to choose the direction of rotation of the DC motor. Then, for the output ports we have the PWM\_OUT\_H allowing to turn the motor clockwise, the PWM\_OUT\_A allowing to turn the motor anticlockwise. But also the PWM\_OUT\_O which is our pwm output signal regardless of clockwise or counter-clockwise direction. And finally, a 4-bit std\_logic\_vector signal called LED\_OUT to control the LED display.

```
entity PWM_Generator_50M is

port (
    clk_50M: in std_logic; -- 50MHz clock input
    DUTY_INCREASE: in std_logic; -- button to increase duty cycle by 10%
    DUTY_DECREASE: in std_logic; -- button to decrease duty cycle by 10%
    sens_H: in std_logic; -- button to drive motor of clockwise
    PWM_OUT_H:out std_logic; -- rotation motor of clockwise
    PWM_OUT_A:out std_logic; -- rotation motor of anticlockwise
    PWM_OUT_O:out std_logic; -- rotation motor
    LED_OUT: out std_logic_vector(3 downto 0) -- gestion des leds
    );
end FWM_Generator_50M;
```

Figure 1 – PWM generator 50 M

- The component DFF\_debounce as well as the process debouncing The DFF\_Debounce.vhd file is an implementation of the D flip-flop to handle bounce when pressing buttons (entries).

```
□ architecture Behavioral of PWM_Generator_50M is

-- D-Flip-Flop for debouncing module

□ component DFF_Debounce
□ Port(

CLK : in std_logic;
    en : in std_logic;
    D : in std_logic;
    Q : out std_logic

end component;
```

Figure 2 – Debounce

Indeed, when the FPGA buttons are pressed, there are unexpected bouncebacks that are generally undesirable in the process operation, the Debouncing process allows to manage its anti-bounce hazards on the FPGA buttons by generating only one pulse with an input clock period when the FPGA button is still pressed, held long enough and released. It creates a delay between two readings.

```
□ process(clk_50M)
| begin
□ if(rising_edge(clk_50M)) then
| counter_slow <= counter_slow + x"0000001";
| --if(counter_slow>=x"BEBC20") then -- for running on FPGA -- comment when running simulation
□ if(counter_slow>=x"0000001") then -- for running simulation -- comment when running on FPGA counter_slow <= x"0000000";
| end if;
| end if;
| end process;
| end process;
| end process;
| end process</pre>
```

Figure 3 – Clock process

Stage0 and stage1 implement two D flip-flops whose "Duty\_increase" button is connected to the input of the first one and its output is connected to the input of the second one. The output of 2 nd will then be used as the processed information of the button. This is for bounce management and input button signal shaping. The same applies to stage3 and stage4.

```
stage0: DFF_Debounce port map(clk_50M,slow_clk_en , DUTY_INCREASE, tmp1);
stage1: DFF_Debounce port map(clk_50M,slow_clk_en , tmp1, tmp2);
duty_inc <= tmp1 and (not tmp2) and slow_clk_en;

-- debounce part for duty decreasing button ( géré les antis rebond décrémentation)
stage2: DFF_Debounce port map(clk_50M,slow_clk_en , DUTY_DECREASE, tmp3);
stage3: DFF_Debounce port map(clk_50M,slow_clk_en , tmp3, tmp4);
duty_dec <= tmp3 and (not tmp4) and slow_clk_en;</pre>
```

Figure 4

#### - Counter\_pwm, Duty\_cycle, pwm\_out

We want to create a pwm signal from our 50MHz clock with a period equivalent to 10 clock periods. Since the period of the 50MHz clock signal lasts 20ns, the period of our pwm signal will last 200 ns. For this, we need a counter that will count the number of clock signals that will constitute the pwm signal. So we created a 4-bit std\_logic\_vector signal, all initiated at 0, called counter\_PWM. Thanks to a process that is triggered at each movement of the 50M clock, the value of the counter\_pwm will be increased by 1 at each rising edge of the clock. Since we want a pwm of 10 clock periods, we need to limit the counter\_pwm to 9(0 to 9 = 10 periods). For this, if the current value of the counter\_pwm is equal to or greater than 9, the counter\_pwm takes the value 0.

Figure 5 – counter process

Now that the pwm period is set, we need to manage the pwm duty\_cycle (set its width) and thus the duration in high position (= 1) in the pwm period. To do this, we create the DUTY\_CYCLE signal which is std\_logic\_vector of 4 bits but initiated at hexadecimal value 3. In order that the value of the DUTY\_Cycle does not vary within a pwm signal period, we have created the PWM\_Cycle signal which takes the value 1 when the counter\_PWM is equal to 9. This signal therefore marks the end of a pwm period. We are therefore going to create a process that is triggered each time the PWM signal managing the DUTY\_CYCLE changes. In this process, at each rising edge of the PWM\_Cycle signal announcing the end of a pwm period, DUTY\_CYCLE will increase by 1 when DUTY\_INCREASE is equal to 1 or will decrease by 1 when DUTY\_DECREASE is equal to 1. However, as the pwm signal has 10 clock periods, we need to limit the DUTY\_CYCLE signal to 10(=100% of the period), for this the DUTY\_CYCLE must only increase by 1 by the DUTY\_INCREASE when the DUTY\_CYCLE is less than or equal to 10. Likewise, since a DUTY\_CYCLE being less than zero is not possible, we will only allow the DUTY\_CYCLE to be reduced when it is greater than or equal to 1.

FIGURE 6 – PWM Cycle

Thanks to the counter\_PWM and DUTY\_CYCLE signals, we can now get our pwm signal by defining PWM\_OUT as 1 only when the counter\_PWM is less than the DUTY\_CYCLE, otherwise it is 0 . Since the pwm signal period corresponds to 10 clock signal periods and DUTY\_CYCLE is limited to 10 and varies in steps of 1 , each variation of DUTY\_CYCLE therefore corresponds to the variation of the pwm (duty cycle) width by 10%.

```
- PWM \, OUT \, H and PWM \, OUT \, A
```

In order to be able to run our motor in one direction or the other, our server must have two outputs. So, to make the motor turn in one direction, you have to send the pwm to the first output and ground the second output. And conversely, to make it rotate in the other direction, the pwm signal must be placed in the second output and the first output must be grounded. For this in the vhdl code it's simple, we need two output signals PWM\_OUT\_A (counterclockwise) and PWM\_OUT\_H (clockwise) as well as the direction\_H signal (button). When the direction\_H is equal to the output the pwm signal is sent in PWM\_OUT\_A while when it is equal to 0 the signal is sent in PWM\_OUT\_H.

```
PWM_OUT_H <= PWM_OUT when sens_H ='1' else '0';
PWM_OUT_A <= PWM_OUT when sens_H ='0' else '0';
PWM_OUT_O <= PWM_OUT;</pre>
```

FIGURE 7 – PWM OUT

#### - LED application

The application will simply indicate the variation of the duty cycle to the user by displaying on the leds of the DE1 board Altera the number of clock periods that the width of the pwm constitutes. For this, a 4-bit output port called LED\_OUT has been defined as well as an inter LED\_Counter signal also defined as a 4-bit std\_logic\_vector and all set to zero.

In a process, the LED\_Counter will be increased by 1 in hexadecimal at each rising edge of the clock when PWM\_OUT is equal to 1, and when counter\_PWM is greater than or equal to 9 (end of the pwm period), the LED\_Counter returns to zero. LED\_OUT continuously takes the value of this counter and will be displayed on the leds of the board.

```
process(clk_50M)
begin
   if(rising_edge(clk_50M)) then
      if(counter_PWM >= x"9") then
        LED_Counter <= x"0";
      elsif(PWM_OUT = '1') then
        LED_Counter <= LED_Counter + x"1";
   end if;
   end if;
end process;
LED_OUT <= LED_Counter;</pre>
```

Figure 8 – Led process

#### 0.2 Sofware

It should be noted that the command on the certe is not done using the buttons but instructions which are sent from a code written in C language.



FIGURE 9 – Wire setting

Since the system is controlled from the C language, we will first adjust the output wires which will be connected to the interface linked to the card.

Figure 10 – Pin setting

Here we choose the pins on which the wires will be connected the software output wires

FIGURE 11 - Command

```
//write into register to test the adder
*(uint32_t *)h2p_lw_DUTY_INCREASE_addr = 1;
*(uint32_t *)h2p_lw_DUTY_DECREASE_addr = 0;
*(uint32 t *)h2p lw SENS H addr = 1;
*(uint32_t *)h2p_lw_DUTY_INCREASE_addr = 0;
*(uint32 t *)h2p lw DUTY DECREASE addr = 0;
*(uint32 t *)h2p lw SENS H addr = 1;
*(uint32_t *)h2p_lw_DUTY_INCREASE_addr = 0;
*(uint32 t *)h2p lw DUTY DECREASE addr = 1;
*(uint32 t *)h2p lw SENS H addr = 1;
*(uint32_t *)h2p_lw_DUTY_INCREASE_addr = 0;
*(uint32 t *)h2p lw DUTY DECREASE addr = 0;
*(uint32 t *)h2p lw SENS H addr = 1;
*(uint32 t *)h2p lw DUTY INCREASE addr = 1;
*(uint32_t *)h2p_lw_DUTY_DECREASE_addr = 0;
*(uint32_t *)h2p_lw_SENS_H_addr = 0;
*(uint32 t *)h2p lw DUTY INCREASE addr = 0;
*(uint32_t *)h2p_lw_DUTY_DECREASE_addr = 0;
*(uint32 t *)h2p lw SENS H addr = 0;
*(uint32_t *)h2p_lw_DUTY_INCREASE_addr = 0;
*(uint32 t *)h2p lw DUTY DECREASE addr = 1;
*(uint32_t *)h2p_lw_SENS_H_addr = 0;
*(uint32 t *)h2p lw DUTY INCREASE addr = 0;
*(uint32 t *)h2p lw DUTY DECREASE addr = 0;
*(uint32 t *)h2p lw SENS H addr = 0;
```

FIGURE 12 – C intructions

This is the part of code C which allows the duty cycle to increase and decrease by keeping the sens\_H constant each time. This is what will be simulated in the test bench. A suitable loop should be able to vary the duty cycle up to its maximum value and then decrease it to 0.

#### 0.3 TEST BENCH

To be able to test the VHDL code generating a pwm signal, it is necessary to create a file to build the test signals called the test bench. The input and output signals must be defined in this one. The same signals as the pwm signal generating entity have been chosen to match the pwm signal generating entity. The names must not be identical but the number of inputs and outputs must match.

```
COMPONENT PWM_Generator_50M

PORT(

clk_50M: in std_logic; -- 50MHz clock input

DUTY_INCREASE: in std_logic; -- button to increase duty cycle by 10%

DUTY_DECREASE: in std_logic; -- button to decrease duty cycle by 10%

sens_H: in std_logic; -- button to drive motor of clockwise

PWM_OUT_H:out std_logic; -- rotation motor of clockwise

PWM_OUT_A:out std_logic; -- rotation motor of anticlockwise

PWM_OUT_O:out std_logic; -- rotation motor

LED_OUT: out std_logic_vector (3 downto 0) -- gestion des leds

);

END COMPONENT;
```

FIGURE 13 – Component

The signals must then be initialized. The input signals (clk\_50M, DUTY\_INCREASE, DUTY\_DECREASE, H-direction) are set to '0'. The output signals are not assigned to a start value.

The next step is the initialization of the unit under test called UUT (Unit Under Test). The vhdl code you want to test must be called and the test bench signals must be matched to those of the UUT.

FIGURE 14 – Instantiation

Next comes the definition of the process for creating the clock signal. A constant called clk\_period was initialized at 20 ns (corresponding to 50MHz). In this process, the clock (clk\_50M) is set to '0' during half of clk\_period (10ns) and to '1' during the other half. This process therefore creates a square wave signal with a period of 20 ns. The next step is the creation of the simulation process itself. Indeed, the input variables (other than the clock) are defined in this process. For the tests presented in the "Simulation" part, the DUTY INCREASE changes value each time the value of PWM\_OUT\_O changes. This means that the DUTY\_INCREASE is activated every pwm period, which increases the DUTY\_CYCLE every pwm period. This operation is repeated 4 times (4 duty cycle increases of 10%). Then the DUTY\_INCREASE is left at '0' and the same operation is performed with DUTY\_DECREASE. The behavior is similar except that DUTY\_CYCLE decreases.

```
stim_proc: process
begin
                                                              -- duty decrease
             -- duty increase
                                              DUTY_DECREASE <= '1';
                                                   wait until PWM_Out_0 = '1';
DUTY_INCREASE <= '1';
                                               DUTY_DECREASE <= '0';
   wait until PWM_Out_O = '1';
                                                             wait until PWM_Out_O = '0';
DUTY_INCREASE <= '0';
                                              DUTY_DECREASE <= '1';
            wait until PWM_Out_O = '0';
                                                   wait until PWM_Out_0 = '1';
DUTY_INCREASE <= '1';
                                               DUTY_DECREASE <= '0';
   wait until PWM_Out_O = '1';
                                                             wait until PWM_Out_O = '0';
DUTY_INCREASE <= '0';
            wait until PWM_Out_O = '0';
                                              DUTY_DECREASE <= '1';
DUTY_INCREASE <= '1';
                                                   wait until PWM_Out_O = '1';
   wait until PWM_Out_0 = '1';
                                               DUTY_DECREASE <= '0';
DUTY_INCREASE <= '0';
                                                             wait until PWM_Out_O = '0';
            wait until PWM_Out_O = '0';
                                             DUTY_DECREASE <= '1';
DUTY_INCREASE <= '1';
                                                   wait until PWM_Out_O = '1';
   wait until PWM_Out_O = '1';
                                              DUTY_DECREASE <= '0';
DUTY_INCREASE <= '0';
                                                            wait until PWM_Out_O = '0';
            wait until PWM_Out_O = '0';
```

FIGURE 15 - Test Bench

This process simulates the use of the Altera card.Indeed, the control of the DUTY\_INCREASE, DUTY\_DECR and sens\_H signal values simulates the action of the software.

#### 0.4 Simulation

#### **Basics**



Figure 16 – Basic

Here's the first 400 nanoseconds of our pwm generator test bench simulation. We can already see on these first 400 nanoseconds that, as previously explained, the pwm signal has a period equivalent to 10 clock periods and that the counter pwm is counting these 10 periods from 0 to 9. The PWM Cycle signal is at 1 when counter pwm is at 9, thus announcing the end of the pwm period. Thanks to this PWM Cycle, DUTY CYCLE change takes its new value at the very beginning of the pwm period so obviously the DUTY INCREASE or the DUTY DECREASE are changed to 1. In this case, the initial DUTY CYCLE is 3 and after a DUTY INCREASE which was changed to one during the first pwm period, the DUTY CYCLE is now changed to 4. Similarly, after a second DUTY INCREASE the DUTY CYCLE now goes to 5. We also know that pwm is defined so that PWM OUT = 1when counter pwm is less than DUTY CYCLE. And so by comparing these three signals in the simulation we see that it works properly. Indeed, PWM OUT O is at 1 during the first 3 clock periods (30% duty cycle) and switches from the moment when counter pwm is equal to the DUTY CYCLE. In the same way for the second period of the pwm, the DUTY CYCLE has changed to 4 and we have PWM OUT O which is at 1 during the first 4 periods of the clock (counter pwm from 0 to 3). We can also observe that over the first 400 nanoseconds, direction H is at 0 which means that the pwm signal is sent in PWM OUT A (anti clockwise) and therefore PWM OUT H is at 0 all along. Finally, the last two are intended for the application and again everything is working properly. Indeed, LED counter counts all clock periods where the pwm signal is equal to 1 and LED OUT displays this counter.

#### 0.4.1 DUTY INCREASE



FIGURE 17 – Duty increase

On this second simulation image, we see this time entirely the first 4 periods of the pwm signal (up to 800ns) and the beginning of the 5 th period. We observe that during this time labs, there were 3 DUTY\_INCREASE consecutively and that each time the DUTY\_CYCLE increased by 1 at the beginning of the pwm period.

We placed markers at each rising and falling edge of the PWM\_OUT\_O signal in order to be able to measure the different widths of the pwm signal.

During the first period of the pwm, the width is 50 ns. In reality, it should have been 60 ns since the Duty cycle is 30% (DUTY\_CYCLE = 3). This difference can be explained by the fact that the clock starts with a half period and thus 10 ns instead of its 20 nanosecond period. Then on the second period the pwm has a width of 80 ns, the fourth a width of 100 ns and the fifth a width of 120 ns. This corresponds to a duty cycle of 40%, then 50% and finally 60%.

#### 0.4.2 DUTY DECREASE



Figure 18 – Duty decrease

We can see from these two images that after the 3 DUTY\_INCREASE, the test bench is followed by 4 DUTY\_DECREASE. Once again the simulation shows us that everything works, after each DUTY\_DECREASE the DUTY\_CYCLE decreases by 1 and therefore the width decreases each time by 10% The first pwm period is 100 ns wide (50%), the second is 80 ns wide (40%), the third is 60 ns wide (30%) and the fourth is 40 ns wide (20%). This means that each time we have one period less and therefore a 10% reduction in the duty cycle.

#### SENS H



Figure 19 – Sens H

In this image, we observe the first variation of direction\_H in the test bench. We see that from this passage to 1 of direction\_H the pwm signal is well sent in  $PWM\_OUT\_H$  while  $PWM\_OUT\_A$  is now all the time at 0 .

- Tmp1, tmp2, tmp3, tmp4 + slow\_clk\_en + counter\_slow As explained above, the debounce function is designed to handle bounces when the DUTY\_INCREASE and DUTY\_DECREASE buttons are pressed. The following codes are intended to achieve this.

```
-- debounce part for duty increasing button( géré les antis rebond incrémentation)

stage0: DFF_Debounce port map(clk_50M,slow_clk_en , DUTY_INCREASE, tmp1);
stage1: DFF_Debounce port map(clk_50M,slow_clk_en , tmp1, tmp2);
duty_inc <= tmp1 and (not tmp2) and slow_clk_en;

-- debounce part for duty decreasing button ( géré les antis rebond décrémentation)

stage2: DFF_Debounce port map(clk_50M,slow_clk_en , DUTY_DECREASE, tmp3);
stage3: DFF_Debounce port map(clk_50M,slow_clk_en , tmp3, tmp4);
duty_dec <= tmp3 and (not tmp4) and slow_clk_en;
-- for controlling duty cycle by these buttons
```

FIGURE 20 – Anti rebonds

It is desired that when the DUTY\_INCREASE or DUTY\_DECREASE button is pressed, only one activation occurs. To do this, we want that after the DUTY\_INCREASE signal goes to 1 (pressing the botton), the duty\_inc goes to 1 only during the high level of the slow\_clock\_en and between tmp\_1 and tmp\_2 (which differs from a clock duration of the slow\_clock\_en). The principle is the same for the duty\_decrease. We can see from these simulation captures that the debounce function works correctly. Indeed, duty\_inc or duty\_dec will each time switch to 1 during a slow\_clock\_up following the activation of DUTY INCREASE or DUTY\_DECREASE.



Figure 21

#### 0.5 Conclusion

We have adapted the code so that it can work with a 50MHz clock (corresponding to the DE1 Altera board) instead of the 100MHz initially planned. We have also made the necessary modifications to obtain the signals needed to control the DC motor clockwise and anticlockwise. Finally, we have implemented a small application to display in real time on the LEDs of the DE1 Altera board the number of clock cycles that composes the PWM signal width. Simulations of the project could be carried out by the Intel Quartus Prime software via ModelSim. Unfortunately, we were unable to completly achieve the project.